home *** CD-ROM | disk | FTP | other *** search
/ MacWorld: Complete Mac Interactive / Macworld Complete Mac Interactive CD)(1994).iso / The Best of BMUG / Power Mac / ArsMagna PPC / ars.c < prev    next >
Text File  |  1994-06-06  |  46KB  |  1,316 lines

  1. /*
  2.    Copyright (c) Michael S. Morton
  3.      1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993
  4.    All rights reserved.
  5.  
  6.    Permission is granted to anyone to use this software for any purpose on
  7.    any computer, and to alter it and redistribute it freely, subject
  8.    to the following restrictions:
  9.  
  10.    0. This software is provided "as is" and without any express or implied
  11.       warranties, including, without limitation, the implied warranties of
  12.       merchantability and fitness for a particular purpose.
  13.  
  14.    1. The author is not responsible for the consequences of the use of this
  15.       software, no matter how awful, even if they arise from flaws in it.
  16.  
  17.    2. The origin of this software must not be misrepresented, either by
  18.       explicit claim or by omission.  The credits must mention that this
  19.       software is based on the algorithm as described in the November 1987
  20.       issue of BYTE Magazine. The credits must appear in the About... dialog
  21.       and any documentation.
  22.  
  23.    3. Altered versions must be plainly marked as such, and must not be
  24.       misrepresented as being the original software.  The credits must appear
  25.       in the About... dialog and any documentation.
  26.  
  27.    4. This software may not be sold except as part of a substantially
  28.              different program.
  29.  
  30.      5. The Boston Museum of Science is expressly forbidden to use this program
  31.              for profit or to produce revenue.
  32.  
  33.    6. This notice may not be removed or altered.
  34. */
  35.  
  36. /*    [Set tabs to 2 characters.]
  37.  
  38.         This is the source for Ars Magna v1.1, a version with some improvements but
  39.         at least one serious bug.  It was never widely released.
  40.  
  41.         This source is not up-to-date, except for this comment.
  42.  
  43.         I plan to return to work on anagram engines and user interfaces someday, but
  44.         it's not going to be soon.  I've been thinking hard about user interfaces
  45.         for this app since 1984 (the original app was done in '82), and haven't come
  46.         up with anything good.  My hope is that various developers will experiment,
  47.         steal each other's best ideas, and in general have a real good time.  I've
  48.         already seen some people take Ars Magna's bad user interface and produce a
  49.         worse one; get creative here, folks, will ya?
  50.  
  51.         A few disconnected ideas, more or less off the top of my head:
  52.  
  53.             * I want to be able to say "Show me everything with word ____"
  54.             * Then, of course, the remaining mainstream anagrams omit that word
  55.             *    Similarly, I want to be able to say "Show me everything with words ____
  56.                 and ____".
  57.             * I want to be able to say "Don't ever show me this word during this session"
  58.             * I want to be able to say "Don't ever show me this word, ever"
  59.             * This version tries to do one-word anagrams, then two-word, etc.  It
  60.                 tends to lock up at the each length; there's some obvious pruning
  61.                 that can be done here.
  62.             *    Reviewing the output can take hours, so:
  63.                 (1) the user should be able to do nearly everything with the keyboard
  64.                 (2) they should be able to save their complete search state
  65.             * The app should handle various dictionary formats, and set one or more
  66.                 default dictionaries as a preference.
  67.             * I've never been able to find foreign-language dictionaries; anyone
  68.                 know if there are public-domain ones around?
  69.             *    Word order and search order are very important to producing good
  70.                 anagrams early in the search.  What are the possible orderings?
  71.                 Which help in which ways?  How should the user control them?
  72.             * The list of eligible words should be displayed somewhere.  The user
  73.                 should be able to save it, specify hot words to try first, etc.
  74.             *    There should be a way to show incomplete anagrams, with the user
  75.                 specifying the maximum number of unused letters to show (zero would
  76.                 make the app behave as it does now).  They should be able to rearrange
  77.                 unused letters under keyboard or mouse control.
  78.             *    Once you've found an anagram which looks interesting, the app should
  79.                 let you rearrange them under keyboard or mouse control, or quickly
  80.                 display all orderings of the words.  Of course, a quick command should
  81.                 save an anagram to the list of cool anagrams.
  82.             *    When listing anagrams, a change in a given column should be highlighted
  83.                 to wake up a bored user.  This version does this crudely with capitals
  84.                 when a word first appears in a column, as in this example:
  85.                         CLIMATE HO MR NO
  86.                         climate ho mr ON
  87.                         climate MR NO OH
  88.                         climate mr OH ON
  89.             *    I hate retyping anagrams for posting or publication, so the app should:
  90.                 (1) Allow easy word-shuffling and punctuation and capitalization
  91.                         changes in the saved list of cool anagrams.
  92.                 (2) Quickly check a text file or the clipboard's contents to make
  93.                         sure they're right.
  94.             *    You should be able to export the list of cool anagrams to the
  95.                 clipboard or a text file.
  96.             * The current bit-banging scheme makes sure that an anagram contains
  97.                 exactly the right number of letters, by adding them up and checking
  98.                 that the count is exactly right.  This could be generalized to make
  99.                 sure the count is in a certain range for an attribute OTHER than a
  100.                 letter.  This is useful for:
  101.                 (a) I'm using several dictionaries, but only one is new, so show me
  102.                         only anagrams with one or more words from that dictionary.
  103.                 (b)    I want to see no more than 2 one-letter words.
  104.                 (c)    I want to see at least one word that I've marked as interesting.
  105.             *    Or, if you want to use the counter scheme for "no more than one of"
  106.                 or "at least one of", you could use AND and OR instead of addition.
  107.             *    I should be able to quickly count all the anagrams which will result
  108.                 from a partial anagram I've found, to help me decide when to explore.
  109.             *    I've tried a NeXT-style browser interface and, at least the way I did
  110.                 it, it was lousy.  Not interactive enough.
  111.             *    You might be able to prune or order anagrams on the basis of available
  112.                 or constructed information on word-pair frequencies in natural languages.
  113.                 This idea strikes me as promising, but I haven't done anything with it.
  114.             *    Extra credit: Parse the anagrams as natural language to find good ones.
  115.  
  116.             Don't expect that doing all of the above will give you a good app.  This is
  117.             just a laundry list, not an overall vision of a good UI.  I've talked to
  118.             good anagrammers, and they work on paper and in their heads -- devising an
  119.             interface which is easier for them, not just for a beginner, will be hard.
  120.  
  121.         If you develop an app, prototype or not, commercial or not, I'd be interested
  122.         in seeing it and commenting on the UI.  If you feel like sharing the source,
  123.         I'd be interested in that, too.
  124.  
  125.         Keep in touch,
  126.                 Mr. Machine Tool
  127.                 February, 1993
  128.  
  129.         Email:    Mike_Morton@Proponent.com
  130.         Paper:    Mike Morton, POB 11299, Honolulu, HI  96828
  131. */
  132.  
  133. #include <stdio.h>
  134. #include <stdlib.h>
  135. #include <string.h>
  136. #include <console.h>
  137. #include <Types.h>
  138. #include <Files.h>
  139.  
  140. /* Easily changeable constants: */
  141. #define STRMAX            100                                /* size of some strings */
  142. #define MAXFILES        10                                /* maximum number of dictionaries */
  143. #define MAXMASKS        10                                /* number of "masks" for bitbanging */
  144.                                                         /* note that doanagram() depends on MAXMASKS */
  145. #define DFTFILE            "anagram dict"                    /* default dictionary name */
  146. #define    WPTRDELTA        512                                /* # of word ptrs to add */
  147. #define TEXBSIZE        2048                            /* size of a block of text */
  148. #define DOTFREQ            25                                /* anagrams per dot, when filing/counting */
  149. #define DOTSPERLINE        40                                /* dots per line for filing/counting */
  150. #define PAGESIZE        22                                /* lines between prompts */
  151. #define WORDCHUNK        5000                            /* size of chunks for words */
  152.  
  153. /* Not-so-easily-changeable things: */
  154. #define mask        long                                    /* masks are stored in longwords */
  155. #define maskwidth    8*sizeof(mask)            /* and their size is important */
  156.  
  157. /* Global stuff -- phrase information: */
  158. char phrase [STRMAX];                                    /* the phrase to be anagrammed */
  159. char origphrase [STRMAX];                            /* unmunged copy of phrase */
  160. short freqs [26];                                            /* frequency table for phrase */
  161. short letmask [26], letbit [26];            /* mask, bit positions for each letter */
  162. short letwidth[26];                                        /* width in mask of each letter */
  163. short usedmasks;                                            /* count of used masks */
  164. short minWords, maxWords;
  165.  
  166. mask oflodesc [MAXMASKS];                            /* descriptor used to detect overflow */
  167. mask phrasedesc [MAXMASKS];                        /* descriptor for main phrase */
  168. mask startdesc [MAXMASKS];                        /* starting descriptor for search */
  169.  
  170. /* Global stuff -- Dictionary information */
  171. short numdicts = 0;                                        /* number of dictionaries */
  172. FILE *dicts [MAXFILES];                                /* dictionary files */
  173. short numwords;                                                /* number of eligible words */
  174. short maxwords = 0;                                        /* number of words which fit in *wordlist */
  175. char **wordlist = NULL;                                /* words[0..maxwords-1],
  176.                                                                                             with 0..numwords-1 used */
  177. mask *worddescs;                                            /* descriptors [0..numwords-1] */
  178. struct wchunk {
  179.     struct wchunk *next;
  180.     char text [WORDCHUNK];
  181. };
  182. char *nextwcp = NULL;                                    /* pointer to next char to fill */
  183. short nextwleft = 0;                                    /* chars left in this block */
  184. struct wchunk *curchunk = NULL;                /* pointer to current chunk */
  185. struct wchunk *chunkhead = NULL;            /* pointer to first chunk */
  186.  
  187. /* Global stuff -- For printing anagrams */
  188. char *anawords [20];                                    /* stacked words */
  189. char **anaptr;                                                /* pointer to first unused slot */
  190. FILE *anafile;                                                /* stdin for screen, output file, or NULL */
  191. short silent;                                                    /* 1 iff not printing to screen */
  192. unsigned long anacount;                                /* number of anagrams found */
  193. unsigned long startt, endt, tottime;    /* timing information */
  194. short paging;                                                    /* pause after each screenful? */
  195.  
  196. enum {                                                                /* types of help */
  197.     dict1, dict1dft, dict2,
  198.     anahelp,
  199.     scf,
  200.     overadd,
  201.     quithelp,
  202.     helppage, helppaused
  203. };
  204.  
  205. #define DR register                                        /* Debugging Register */
  206. #define pf printf
  207.  
  208. void dophrase (void);
  209. void clean (char *s);
  210. void cleannl (char *s);
  211. void makefreqs (char *s);
  212. short findbits (void);
  213. short findbitwidth (short count);
  214.  
  215. void makeof (void);
  216. short makedesc (char *str, mask desc[]);
  217. short okword (char *str);
  218. short getdicts (void);
  219. short getwords (void);
  220. short storeword (char *word);
  221. void freewords (void);
  222. short mycmp (char **s1p, char **s2p);
  223. void sortwords (void);
  224. void nodups (void);
  225. void printwords (void);
  226. void makeWrite (FILE *f);
  227. short makemasks (void);
  228. void freemasks (void);
  229. short doanagrams (short curword, mask *curstate, short wordsLeft);
  230. short printana (void);
  231. short exitcheck (void);
  232. void getoutput (void);
  233. short openoutfile (char *name);
  234. short getline (char *prompt, char *keywords, short keyneeded, char *dft, char *reprompt,
  235.                         short helpstate, char *response);
  236. void dohelp (short state);
  237. short keyfind (char *words, char *oneword);
  238. short findstr (char *master, char *sub);
  239. short strNequal (char *s1, char *s2, short size);
  240. long msec (void);
  241. void starttimer (void);
  242. void stoptimer (void);
  243. void inittimer (void);
  244. void printtime (void);
  245. void chkerr (short i);
  246.  
  247. void main (void)
  248. {
  249.     //cgotoxy (0, 0, stdout);                        /* don't scroll initial stuff */
  250.     pf (" \n");
  251.     pf ("********************************************************************************\n");
  252.     pf ("*      A R S  M A G N A (TM), an anagram generator, version 1.1 (24 Sep '90)   *\n");
  253.     pf ("* Copyright 1986-1993 Michael Morton (aka 'Mr. Machine Tool', 'Harmonic motel')*\n");
  254.     pf ("*                            All rights reserved.                              *\n");
  255.     pf ("*    Based on my algorithm as described in the November 1987 issue of BYTE.    *\n");
  256.     pf ("*                                                                              *\n");
  257.     pf ("* Ported to Think C 5.0.4 by Matthew Xavier Mora              (9-21-93)        *\n");
  258.     pf ("*                            mxmora@unix.sri.com                               *\n");
  259.     pf ("* Ported to MWC DR3       by Matthew Xavier Mora              (6-06-94)        *\n");
  260.     pf ("*                            mxmora@unix.sri.com                               *\n");
  261.     pf ("********************************************************************************\n");
  262.     pf ("\n");
  263.     pf ("QUICK...This program finds anagrams for a phrase or name.  You can just press\n");
  264.     pf (" INFO   RETURN for most questions, or type 'HELP' at any point for more info.\n");
  265.     pf ("\n");
  266.  
  267.     numdicts = getdicts ();                            /* just once: open up dictionaries */
  268.  
  269.     while (1)
  270.     {
  271.         printf ("\n");
  272.         getline ("What's the phrase you'd like to anagram? ",
  273.                             "", 0, "", "\nWhat's the phrase you'd like to anagram? ",
  274.                             anahelp, phrase);
  275.         strcpy (origphrase, phrase);            /* remember it, before we... */
  276.         clean (phrase);                                        /* clean up the string */
  277.  
  278.         if (strlen (phrase) == 0)                            /* nothing there? */
  279.             printf ("Come on, don't be shy...  try something!\n");
  280.         else dophrase ();
  281.  
  282.         if ((anafile != NULL) && (anafile != stdout))
  283.             fclose (anafile);                                /* heavy-handed to put this here... */
  284.     }                                                        /* end of infinite main loop */
  285. }                                                            /* end of main program */
  286.  
  287. void dophrase ()
  288. {
  289.     short i;                                                        /* counter for masks */
  290.     char response [STRMAX];                            /* dummy input buffer */
  291.     short wordCount;
  292.     short keynum;
  293.  
  294.     makefreqs (phrase);                                    /* build the frequency table */
  295.  
  296.     if (findbits () == 0)                                /* choose bit fields; check size */
  297.     {
  298.         printf ("Sorry -- that phrase is too long to handle.\n");
  299.         return;
  300.     }
  301.  
  302.     getoutput();                                                /* where should we put this stuff? */
  303.  
  304.     if (anafile == stdout)
  305.         paging = getline ("Want output to pause after each page [RETURN for YES]? ",
  306.                                             "#no#yes", 1, "yes",
  307.                                             "Please type YES or NO.  Want output to pause after each page? ",
  308.                                             helppage, response);
  309.     else paging = 0;                                        /* no paging if not printing */
  310.  
  311.     makeof ();                                                    /* compute overflow and "full" descriptors */
  312.  
  313.     makedesc (phrase, phrasedesc);            /* make descriptor for phrase; ignore result (can't fail) */
  314.  
  315.     for (i = 0; i <= usedmasks; i++)        /* loop through all masks we used... */
  316.         startdesc[i] =                                        /* and for each one, the starting descriptor... */
  317.             phrasedesc[i];                                    /* ...is the full one */
  318.  
  319.     getwords ();                                                /* read in words for this phrase */
  320.     if (numwords == 0) return;                    /* bag it?  OK */
  321.     sortwords ();                                                /* get 'em in the right order */
  322.     nodups ();                                                    /* eliminate duplicate words */
  323.     printwords ();                                            /* dump 'em to a file */
  324.     if (makemasks () == 0) {
  325.         printf ("Sorry!  Couldn't make Masks.");
  326.         return;            /* make masks for them all -- return on error */
  327.     }
  328.  
  329.     inittimer ();
  330.  
  331.     if (anafile != NULL)
  332.         chkerr (fprintf (anafile, "\n\n*** anagrams for '%s' ***\n\n", origphrase));
  333.     for (wordCount = 1; wordCount < 100; wordCount++)
  334.     {
  335.         anaptr = anawords;                                /* point to base of stack */
  336.         anacount = 0;                                            /* no anagrams found so far */
  337.         doanagrams (0, startdesc, wordCount); /* print (or whatever) all anagrams */
  338.         if (anafile != stdout) printf ("\n");    /* don't print after dots */
  339.         printf ("Anagrams found with %d words: %ld.\n", wordCount, anacount);
  340.         stoptimer ();
  341.         keynum = getline ("   ...more (next size of anagram)... ",
  342.             "#stop", 0, "", "   ...more (next size of anagram)... ",
  343.             helppaused, response);
  344.         starttimer ();
  345.         if (keynum == 0)                                /* bag this anagram? */
  346.             wordCount = 100;                            /* break the loop */
  347.     }
  348.  
  349.     printtime ();
  350.  
  351.     freemasks ();                                                /* ditch the masks */
  352.     freewords ();                                                /* ditch the words */
  353. }                                                                            /* end of dophrase() */
  354.  
  355. /* clean -- Clean up a string in place: map all letters to lowercase and
  356.    discard everything else. */
  357.  
  358. void clean (s)
  359.     DR char *s;                                                    /* UPDATE: string to clean */
  360. {
  361.     DR char *in = s, *out = s;                    /* reading, writing pointers */
  362.     DR char c;                                                    /* working copy of character */
  363.  
  364.     while (c = *in++)                                        /* loop through whole input string */
  365.     {
  366.         if ( (c>='A') && (c<='Z') )                /* uppercase alphabetic? */
  367.             c -= ('A' - 'a');                                /* yup: map it to lower */
  368.         if ( (c>='a') && (c<='z') )                /* after mapping, is it lower case? */
  369.             *out++ = c;                                            /* yup: store it */
  370.     }                                                                        /* end of loop mapping&discarding */
  371.     *out++ = c;                                                    /* store the final null */
  372. }                                                                            /* end of clean() */
  373.  
  374. /* cleannl -- Toss all newlines out of a string, in place. */
  375.  
  376. void cleannl (s)
  377.     register char *s;
  378. {
  379.     register char *in = s;
  380.     register char *out = s;
  381.     register char c;
  382.  
  383.     while (c = *in++)
  384.         if (c != '\n')
  385.             *out++ = c;
  386.     *out++ = '\0';
  387. }
  388.  
  389. /* makefreqs -- Take a phrase and produce a frequency table from it.  The
  390.    phrase has already been cleaned up, so we know it contains only the
  391.    characters 'a'..'z'. */
  392.  
  393. void makefreqs (s)
  394.     DR char *s;                                                    /* INPUT: string to analyze */
  395.     /* GLOBAL OUTPUT: frequency table */
  396. {
  397.     DR short i;                                                    /* traditional loop index */
  398.  
  399.     for (i = 0; i<26; i++)                            /* loop through and initialize... */
  400.         freqs [i] = 0;                                        /* ...the frequency array */
  401.  
  402.     while (*s)                                                    /* while there's more to the string... */
  403.     {
  404.         freqs [*s - 'a'] ++;                            /* incremement slot for this letter */
  405.         s++;                                                            /* and skip to the next character */
  406.     }                                                                        /* end of loop through string */
  407.  
  408. }                                                                            /* end of makefreqs() */
  409.  
  410.  
  411. /* findbits -- Given the frequency table, find the bit position for each
  412.    character found in the original phrase.  Nonexistent letters are
  413.    ignored.  We may run out of room in the masks in doing this; we
  414.    return 0 on failure. */
  415.  
  416. short findbits ()                                            /* zero <=> failure */
  417.     /* GLOBAL INPUT: frequency table */
  418.     /* GLOBAL OUTPUT: whichword[], whichbit[], count of used masks */
  419. {
  420.     short letter;                                                /* letter value (0..25) */
  421.     short curword = 0, curbit = 0;            /* initial bit and word */
  422.     short width;                                                /* bitwidth of letter's field */
  423.  
  424.     for (letter = 0; letter < 26; letter++)    /* loop through all letters */
  425.     {
  426.         if (freqs[letter] != 0)                        /* any occurrences of this letter? */
  427.         {                                                                    /* yes: find where it'll go */
  428.             width = findbitwidth (freqs [letter]); /* how much room does it need? */
  429.             if (curbit+width > maskwidth)     /* too wide for this word? */
  430.             {                                                                /* yes: have to kick into next word */
  431.                 curword++; curbit = 0;                /* go to start of next word */
  432.                 if (curword >= MAXMASKS)             /* no more room? */
  433.                     return (0);                                    /* no more: report failure */
  434.             }                                                                /* end of kicking into next word */
  435.  
  436.             letmask [letter] = curword;            /* remember which word we go in */
  437.             letbit [letter] = curbit;                /* and bit position in the word */
  438.             letwidth [letter] = width;            /* and the width */
  439.             curbit += width;                                /* advance past this bit field */
  440.         }                                                                    /* end of handling letter found in phrase */
  441.     }                                                                        /* end of loop through A..Z */
  442.  
  443.     usedmasks = curword;                                /* remember highest used mask */
  444.     return (1);                                                    /* indicate success */
  445. }                                                                            /* end of findbits() */
  446.  
  447. /* findbitwidth -- Find the number of bits needed to store a single letter
  448.    up to the specified frequency.  Our output looks like this:
  449.            Frequency:        Width (+ one overflow bit)
  450.            0                <undefined>
  451.            1                1 (+ 1)
  452.            2..3            2 (+ 1)
  453.            4..7            3 (+ 1)
  454.    ...etc. */
  455.  
  456. short findbitwidth (DR short count)                        /* find width of field to hold "count" */
  457.                                                 /* INPUT: frequency of letter */
  458. {
  459.     DR short width = 0;                                    /* result */
  460.  
  461.     while (count != 0)                                    /* loop 'til all bits discarded */
  462.     {
  463.         width++;                                                    /* counting the bits... */
  464.         count >>= 1;                                            /* ...and chucking out one more */
  465.     }                                                                        /* end of loop counting bits */
  466.  
  467.     width ++;                                                        /* and one more for the overflow */
  468.     return (width);                                            /* that's the answer */
  469. }                                                                            /* end of findbitwidth() */
  470.  
  471.  
  472. /* makeof -- Find the descriptors for the "overflow" and "full" descriptors.
  473.    The former has the bit just ABOVE each bit field set.  ANDing with this
  474.    set of masks will detect when the thing overflows.  The latter is the
  475.    former minus one in each bit field -- an almost full field. */
  476.  
  477. void makeof ()
  478.     /* GLOBAL INPUT: frequency table, letter information, count of masks */
  479.     /* GLOBAL OUTPUT: oflodesc */
  480. {
  481.     short l;                                                        /* letter number */
  482.     short mnum, bnum, bwidth;                        /* mask #, bit #, field width */
  483.     mask onebit, ovbit;                                    /* bits for single letter, overflow */
  484.     short i;                                                        /* usual counter */
  485.  
  486.     for (i = 0; i <= usedmasks; i++)        /* clean out... */
  487.         oflodesc [i] = 0;                                    /* ...each overflow mask */
  488.  
  489.  
  490.     for (l = 0; l < 26; l++)                        /* loop through letters */
  491.         if (freqs [l] != 0)                                /* any letters in the phrase */
  492.         {
  493.             bnum = letbit [l];                            /* what's the bit # for this letter? */
  494.             bwidth = letwidth [l];                    /* how wide is the field? */
  495.             mnum = letmask [l];                            /* and which mask does field go in? */
  496.  
  497.             onebit = 1;                                            /* start with a right-aligned 1 */
  498.             onebit <<= bnum;                                /* ...align it with the field */
  499.             ovbit = onebit << (bwidth-1);        /* and find where the overflow bit is */
  500.  
  501.             oflodesc [mnum] |= ovbit;                /* put the overflow bit in */
  502.         }                                                                    /* end of handling letter in phrase */
  503. }                                                                            /* end of makeof() */
  504.  
  505.  
  506.  
  507. /* makedesc -- Create the descriptor for a string.  If the string contains
  508.    too many of any letter, we return zero to say so. */
  509.  
  510. short makedesc (str, desc)                        /* returns zero <=> failure */
  511.     register char *str;                                    /* INPUT: string to analyze */
  512.     register mask desc[];                                /* OUTPUT: descriptor for string */
  513.     /* GLOBAL INPUT: frequency table, letter information */
  514. {
  515.     register short l;                                        /* letter number */
  516.     register char c;                                        /* character from the string */
  517.     short sfreqs [26];                                    /* string's frequency profile */
  518.     register short i;                                        /* loop counter */
  519.     register mask b;                                        /* a bit, for ORing into the desc */
  520.  
  521.     if (*str == '\0')                                        /* null string? */
  522.         return (0);                                                /* no good */
  523.  
  524.     for (i = 0; i <= usedmasks; i++)        /* go through all used masks... */
  525.         desc[i] = 0;                                            /* ...initializing their descriptors */
  526.  
  527.     for (l = 0; l < 26; l++)                        /* loop through all letters... */
  528.         sfreqs [l] = 0;                                        /* zeroing their frequency */
  529.  
  530.     while (c = *str++)                                    /* pick up all characters in str... */
  531.         sfreqs [c - 'a'] ++;                            /* ...tallying up their count */
  532.  
  533.     for (l = 0; l < 26; l++)                        /* loop through all letters... */
  534.         if (sfreqs [l] != 0)                            /* did it occur in the string? */
  535.         {
  536.             if (sfreqs [l] > freqs[l])            /* did it occur more than in the phrase? */
  537.                 return (0);                                        /* yes: fail */
  538.             b = sfreqs [l];                                    /* start with the count */
  539.             b <<= letbit [l];                                /* shift it into the field */
  540.             desc [letmask [l]] +=    b;                /* now add it into the mask */
  541.         }                                                                    /* end of handling letter in str */
  542.  
  543.     return (1);                                                    /* we're OK -- say so */
  544. }                                                                            /* end of makedesc */
  545.  
  546. /* okword -- See if a string is eligible for use in anagrams.
  547.    We return zero if not. */
  548.  
  549. short okword (str)                                        /* returns zero <=> failure */
  550.     register char *str;                                    /* INPUT: string to analyze */
  551.     /* GLOBAL INPUT: frequency table */
  552. {
  553.     register short l;                                        /* letter number */
  554.     register char c;                                        /* character from the string */
  555.     short sfreqs [26];                                    /* string's frequency profile */
  556.  
  557.     if (*str == '\0')                                        /* null string? */
  558.         return (0);                                                /* no good */
  559.  
  560.     for (l = 0; l < 26; l++)                        /* loop through all letters... */
  561.         sfreqs [l] = 0;                                        /* zeroing their frequency */
  562.  
  563.     while (c = *str++)                                    /* pick up all characters in str... */
  564.     {
  565.         c -= 'a';
  566.         if (++sfreqs [c] > freqs[c])            /* ...tallying up their count */
  567.             return (0);                                            /* ...and checking each one */
  568.     }
  569.     
  570.     return (1);                                                    /* we're OK -- say so */
  571. }                                                                            /* end of okword */
  572.  
  573. /* getdicts -- Get and open up at least one dictionary. */
  574.  
  575. short getdicts ()                                            /* return count of opened dicts */
  576.     /* GLOBAL OUTPUT: dicts[] */
  577. {
  578.     short count = 0;                                        /* initially none found */
  579.     short done = 0;                                            /* done flag */
  580.     short defaultavail;                                    /* default-file-failed flag */
  581.     char dictname [STRMAX];                            /* dictionary file name */
  582.     FILE *testfile;                                            /* used for checking for default file */
  583.     short helptype;
  584.  
  585.     /* is the default file available? */
  586.  
  587.     testfile = fopen (DFTFILE, "r");
  588.     if (testfile == NULL)                                /* blew it? */
  589.         defaultavail = FALSE;                            /* yup -- remember it's not there */
  590.     else {                                                            /* found it */
  591.         defaultavail = TRUE;                            /* remember this */
  592.         fclose (testfile);                                /* and chuck it */
  593.     }                                                                        /* end of successful find */
  594.  
  595.     while (! done)                                            /* loop 'til we're happy */
  596.     {
  597.         if (count == 0)                                        /* no files open yet? */
  598.         {                                                                    /* yes: special greeting */
  599.             printf ("Name of dictionary file");
  600.             helptype = dict1;                                /* set help state */
  601.             if (defaultavail)
  602.             {
  603.                 printf (" [press RETURN to use '%s']", DFTFILE);
  604.                 helptype = dict1dft;
  605.             }
  606.         }
  607.         else
  608.         {
  609.             printf ("Next dictionary name [press RETURN if no more]");
  610.             helptype = dict2;
  611.         }
  612.         getline ("? ", "", 0, "", "Dictionary name? ", helptype, dictname); /* ask 'em */
  613.  
  614.         if (strlen (dictname) == 0)                /* just RETURN? */
  615.         {                                                                    /* handle untalkative input */
  616.             if (count == 0)                                    /* no files yet? */
  617.             {
  618.                 if (defaultavail == 0)                /* no files yet? */
  619.                     printf ("Come now, don't be shy.\n");    /* no default available */
  620.                 else strcpy (dictname, DFTFILE);    /* use default name */
  621.             }
  622.             else done = TRUE;                                /* one or more files is OK */
  623.         }                                                                    /* end of handling RETURN */
  624.  
  625.         if (strlen (dictname) != 0)                /* did we read (or plug in) a name? */
  626.         {                                                                    /* we got a name... */
  627.             testfile = fopen (dictname, "r"); /* ...try to open it */
  628.             if (testfile == NULL)                        /* failed? */
  629.             printf ("Sorry, can't find the file '%s'.  Try again?\n", dictname);
  630.             else dicts [count++] = testfile;    /* stack this dict; bump count */
  631.         }                                                                    /* end of processing input name */
  632.  
  633.         if (count >= MAXFILES)                        /* full up? */
  634.         {                                                                    /* yes: force an exit */
  635.             printf ("\nThat's enough files -- can't handle any more!\n");
  636.             done = TRUE;
  637.         }
  638.  
  639.     }                                                                        /* end of looping 'til done */
  640.  
  641.     return (count);                                            /* say how many dictionaries we got */
  642. }                                                                            /* end of getdicts() */
  643.  
  644. /* getwords -- Read in all the usable words from all the dictionaries.
  645.    We set the count of usable words, including zero if there are
  646.    none.  If we run out of memory, we return zero.  Either way, we
  647.    print a message so the caller doesn't have to. */
  648.  
  649. short getwords ()                                                        /* we return the count of words */
  650. {
  651.     register short dnum;                                /* dictionary number */
  652.     register FILE *infile;                            /* one dictionary's FILE */
  653.     char word [STRMAX];                                    /* one word from the dictionary */
  654.     char cleanword [STRMAX];                        /* cleaned-up copy */
  655.     register long readcount = 0;                /* count of words read */
  656.     char lastline [STRMAX];                            /* last line read */
  657.     char inpline [STRMAX];                            /* raw input line */
  658.     char *inp;                                                    /* input line pointer */
  659.     short common;                                                /* count of letters in common */
  660.  
  661.     numwords = 0;                                                /* no words yet */
  662.     for (dnum = 0; dnum < numdicts; dnum++)    /* loop through all dictionaries */
  663.     {
  664.         printf ("Reading dictionary...");
  665.         infile = dicts [dnum];                        /* grab this dictionary */
  666.         fseek (infile, 0L, 0);                        /* reset to start of file */
  667.         while (1)                                                    /* EOF is noticed in mid-loop */
  668.         {
  669.             if (fgets (inpline, STRMAX, infile) == NULL)    /* get a word */
  670.                 break;                                                /* if end-of-file, quit */
  671.             common = 0;
  672.             inp = inpline;                                    /* point to start of input line */
  673.             while ( (*inp >= '0') && (*inp <= '9') ) /* process digits... */
  674.                 common = (common * 10) + (*inp++ - '0'); /* ...accumulate number */
  675.             lastline [common] = '\0';                /* take first N chars of last line */
  676.             strcat (lastline, inp);                    /* and add the rest */
  677.  
  678.             strcpy (word, lastline);                /* now get the word */
  679.             cleannl (word);                                    /* ditch the newline */
  680.             strcpy (cleanword, word);                /* make a copy... */
  681.             clean (cleanword);                            /* ...and clean it up */
  682.             readcount++;
  683.             if ((readcount % 1000) == 0)
  684.             {    printf (".");
  685.                 fflush (stdout);                            /* don't let this line buffer up */
  686.             }
  687.  
  688.             if (okword (cleanword) != 0)        /* can it be used in anagrams? */
  689.             {                                                                /* yes! */
  690.                 if (storeword (word) == 0)        /* store it; check for failure */
  691.                     return (0);                                    /* say we can't do it */
  692.             }                                                                /* end of handling useful word */
  693.         }                                                                    /* end of loop through one dict */
  694.         printf ("\n");                                        /* ... left us in mid-line */
  695.     }                                                                        /* end of loop through dictionaries */
  696.  
  697.     if (numwords == 0)                                    /* NOTHING found? */
  698.         printf ("Sorry -- absolutely NO usable words found!\n");
  699.     else printf ("%d usable words found.\n", numwords);
  700.  
  701. }                                                                            /* end of getwords () */
  702.  
  703. short storeword (word)
  704.     register char *word;
  705. {
  706.     register char *memword;                            /* allocated word */
  707.     register short len = 1 + strlen (word);    /* size we need to store word */
  708.     register short newmaxwords;                    /* new array size */
  709.     register short size;                                /* size of new block */
  710.  
  711.     if (len > nextwleft)                                /* no room in this chunk? */
  712.     {
  713.         struct wchunk *newchunk;
  714.  
  715.         newchunk = (struct wchunk *) malloc (sizeof (struct wchunk));
  716.         if (newchunk == NULL)                            /* blew it? */
  717.         {
  718.             printf ("Sorry -- not enough memory for this anagram!\n");
  719.             freewords ();                                        /* discard accumulated words */
  720.             return (0);                                            /* say so */
  721.         }
  722.         newchunk -> next = chunkhead;            /* make this... */
  723.         chunkhead = newchunk;                            /* ...the head chunk */
  724.         curchunk = newchunk;                            /* which (coincidentally) is the current one, too */
  725.         nextwcp = newchunk -> text;                /* point to first usable character */
  726.         nextwleft = WORDCHUNK;                        /* remember available size */
  727.     }
  728.  
  729.     memword = nextwcp;                                    /* point to first free character */
  730.     nextwcp += len;                                            /* skip to next slot */
  731.     nextwleft -= len;                                        /* and debit available stuff */
  732.  
  733.     strcpy (memword, word);                            /* store the word in new chunk */
  734.  
  735.     if ((numwords+1) >= maxwords)                /* will this overflow the current list? */
  736.     {
  737.         newmaxwords = maxwords + WPTRDELTA;    /* jump to next size */
  738.         size = newmaxwords * sizeof (char *); /* find new size */
  739.  
  740.         if (wordlist == NULL)
  741.             wordlist = (char **) malloc (size); /* first time */
  742.         else wordlist = (char **) realloc (wordlist, size); /* grow the block */
  743.  
  744.         if (wordlist == NULL)                            /* blew it? */
  745.         {                                                                    /* looks that way */
  746.             printf ("Sorry -- not enough memory for this anagram!\n");
  747.             free (memword);                                    /* chuck word not yet in list */
  748.             freewords ();                                        /* discard accumulated words */
  749.             return (0);                                            /* say to give up */
  750.         }
  751.  
  752.         maxwords = newmaxwords;                        /* remember new maximum */
  753.     }                                                                        /* end of handling list overflow */
  754.  
  755.     wordlist [numwords++] = memword;        /* stack this word in the list */
  756.  
  757.     return (1);                                                    /* the sweet smell of success */
  758. }
  759.  
  760. void freewords ()
  761. {
  762.     struct wchunk *p;
  763.  
  764.     for (p = chunkhead; p = p -> next; p != NULL)
  765.         free (p);
  766.  
  767.     nextwcp = NULL;
  768.     nextwleft = 0;                                            /* force allocation next time */
  769.     curchunk = NULL;
  770.     chunkhead = NULL;
  771. }                                                                            /* end of freewords() */
  772.  
  773.  
  774. short mycmp (s1p, s2p)
  775.     char **s1p, **s2p;
  776. {
  777.     register char *s1 = *s1p;
  778.     register char *s2 = *s2p;
  779.     register short i;
  780.  
  781.     i = strlen (s2) - strlen (s1);    /* if S2 is longer, S1 is greater */
  782.  
  783.     if (i != 0)
  784.         return (i);
  785.     return (strcmp (s1, s2));
  786. }
  787.  
  788. void sortwords ()
  789. {
  790.     printf ("Sorting words..."); inittimer ();
  791.     qsort ((char *) wordlist, numwords, 4, (void *)mycmp);
  792.     printf ("\n");
  793. }
  794.  
  795. void nodups ()
  796. {
  797.     register short wordnum = 0;
  798.     register short move;
  799.  
  800.     while (wordnum < numwords-1)
  801.     {
  802.         if (strcmp (wordlist [wordnum], wordlist [wordnum+1]) != 0 )
  803.             wordnum ++;                                        /* different: go to next word */
  804.         else {                                                    /* same: have to move down */
  805.  
  806.             for (move = wordnum; move < numwords - 1; move++)
  807.                 wordlist [move] = wordlist [move+1];
  808.             -- numwords;
  809.         }
  810.     }
  811. }
  812.  
  813. void printwords ()
  814. {    char response [STRMAX];
  815.     FILE *f;
  816.     register short wordnum = 0;
  817.  
  818.     printf ("What file do you want the word list in [RETURN if none]? ");
  819.     gets (response);                                    /* get a line of input */
  820.     if (strlen (response) == 0)                /* pressed RETURN? */
  821.         return;                                                    /* yes: bag it */
  822.  
  823.     f = fopen (response, "a");                /* open, append */
  824.     if (f == NULL)
  825.     {    printf ("Can't output to that file; sorry!");
  826.         return;
  827.     }
  828.  
  829.     fseek (f, 0L, 2);                                    /* reset to the end */
  830.     while (wordnum < numwords)
  831.     {    fprintf (f, "%s\r", wordlist [wordnum]);
  832.         ++wordnum;
  833.     }
  834.  
  835.     /* makeWrite (f); */                                        /* make it MacWrite-able */
  836.     fclose (f);
  837. }
  838.  
  839. void makeWrite (f)
  840.     FILE *f;
  841. {    short refnum;
  842.     FInfo info;
  843.     Str255 s;
  844.  
  845.     //refnum = f->refnum;
  846.     GetFInfo ((StringPtr)&s, refnum, &info);
  847.     info.fdType = 'TEXT';
  848.     info.fdCreator = 'MACA';
  849.     SetFInfo ((StringPtr)&s, refnum, &info);
  850. }
  851.  
  852. short makemasks ()
  853.     /* GLOBAL INPUT: numwords, wordlist */
  854.     /* GLOBAL OUTPUT: worddescs[] */
  855. {
  856.     short i;
  857.     short onesize;
  858.     mask *descp;
  859.     char wordcopy [STRMAX];
  860.     long size;
  861.     
  862.     onesize = (usedmasks+1) * sizeof (mask);
  863.     size = (long) numwords * onesize;                    /* compute size of all descriptors */
  864.  
  865.     worddescs = (mask *) malloc (size);    /* find space for it */
  866.     if (worddescs == NULL)    {                        /* blew it? */
  867.         printf("Can't Alocate %ld bytes.",size);
  868.         return (0);
  869.     }
  870.     descp = worddescs;                                    /* point to the first descriptor slot */
  871.     for (i = 0; i < numwords; i++)            /* loop through every word... */
  872.     {
  873.         strcpy (wordcopy, wordlist [i]);    /* ...copy the word */
  874.         clean (wordcopy);                                    /* ...clean it up */
  875.         makedesc (wordcopy, descp);                /* ...and store each one's descriptor */
  876.         descp += (usedmasks+1);                        /* ...and bump to next slot */
  877.     }
  878.     return (1);                                                    /* no memory problems */
  879. }                                                                            /* end of makemasks() */
  880.  
  881. void freemasks ()
  882. {
  883.     free (worddescs);
  884. }
  885.  
  886. short doanagrams (register short curword,
  887.                  register mask *curstate,
  888.                  short wordCount)                    /* return 0 to halt search */
  889.                                                     /* current word number */
  890.                                                     /* descriptor for current state */
  891.                                                     /* words left to stack before we print anagram */
  892. {
  893.     mask newdesc [MAXMASKS];                        /* new descriptor */
  894.     register mask newmask;                            /* one mask from descriptor */
  895.     register mask *curdesc;                            /* current word's descriptor */
  896.     register long overflow;                            /* does this combination overflow? */
  897.     register long bitsleft;                            /* is this combination full? */
  898.  
  899.     curdesc = &worddescs [curword * (usedmasks+1)];
  900.  
  901.     wordCount--;                                                /* count the word we're going to stack */
  902.     while (curword < numwords)                    /* loop through all words after this one */
  903.     {
  904.  
  905. #define CASE(MNUM)                                                                                                                    \
  906.         newmask = curstate [MNUM] - curdesc [MNUM];    /* subtract from anagram */    \
  907.         if (overflow =                                        /* remember if there was oflo... */        \
  908.             (newmask & oflodesc [MNUM]))        /* and, if so, bag it */                            \
  909.                 break;                                                 /* note it; break out */                            \
  910.         newdesc [MNUM] = newmask;                    /* it's OK; store it */                                \
  911.         bitsleft |= newmask;                            /* note if done */
  912.  
  913.         bitsleft = 0;                                            /* assume complete 'til we see otherwise */
  914.         overflow = 0;                                            /* no overflow 'til we've seen it */
  915.         switch (usedmasks)
  916.         {
  917.             case 5:    CASE(5)
  918.             case 4:    CASE(4)
  919.             case 3:    CASE(3)
  920.             case 2:    CASE(2)
  921.             case 1:    CASE(1)
  922.             case 0:    CASE(0)
  923.         }
  924.  
  925.         if (! overflow)
  926.         {
  927.             *anaptr++ = wordlist [curword]; /* save this word */
  928.  
  929.             if ((! bitsleft)                                /* no bits left? */
  930.                     && (wordCount == 0))                /* ...and time to print? */
  931.             {
  932.                 if (printana () == 0)                    /* print it; do they want us to stop? */
  933.                   return (0);                                    /* yes: stop */
  934.             }
  935.             else if (wordCount)                            /* dive deeper only if caller wants more words */
  936.             {
  937.                 if (doanagrams (curword, newdesc, wordCount) == 0)
  938.                     return (0);
  939.             }
  940.  
  941.             --anaptr;                                                /* discard the word from the stack */
  942.         }
  943.  
  944.         curword++;                                                /* next word number */
  945.         curdesc += (usedmasks+1);                    /* next word's mask */
  946.     }
  947.  
  948.   return (1);                                                    /* continue the search */
  949. }                                                                            /* end of doanagrams() */
  950.  
  951. static char *lastanawords [20];
  952.  
  953. short printana ()                                            /* return 0 to stop current anagram */
  954. {
  955.     register char **p;
  956.     char response [STRMAX];                            /* dummy buffer for answer */
  957.     short keynum;                                                /* keyword number from getline() */
  958.     Boolean caps;
  959.  
  960.     ++anacount;                                                    /* that's one more anagram */
  961.  
  962.     if (anafile != stdout)                            /* not printing to screen? */
  963.     {
  964.         if ((anacount % DOTFREQ) == 0)        /* every once in a while... */
  965.         {
  966.             printf (".");                                        /* remind them we're here */
  967.             if (exitcheck () == 0)                    /* time to stop THIS anagram? */
  968.                 return (0);                                        /* yes */
  969.             if ((anacount % (DOTFREQ * DOTSPERLINE)) == 0)
  970.             {
  971.                 printf ("anagram #%ld: ", anacount);
  972.                 for (p = anawords; p < anaptr; p++)
  973.                     printf ("%s ", *p);
  974.                 printf ("\n");
  975.             }
  976.         }
  977.     }
  978.  
  979.     if (anafile == NULL) return (1);        /* no file?  no more to do here */
  980.  
  981.     if (! paging)                                                /* no way to stop? */
  982.     {
  983.         if (exitcheck () == 0)                        /* time to stop THIS anagram? */
  984.                 return (0);                                        /* yes */
  985.     }
  986.     else {
  987.         if ((anacount % PAGESIZE) == 0)
  988.         {
  989.             stoptimer ();
  990.             keynum = getline ("   ...more... ",
  991.                 "#stop", 0, "", "   ...more... ",
  992.                 helppaused, response);
  993.             starttimer ();
  994.             if (keynum == 0)                                /* bag this anagram? */
  995.                 return (0);                                        /* yup */
  996.         }
  997.     }
  998.  
  999.     /* Print the damned anagram already: */
  1000.     caps = false;
  1001.     for (p = anawords; p < anaptr; p++)
  1002.     {    if (lastanawords [p-anawords] != *p)    /* different word than showed up last time? */
  1003.             caps = true;
  1004.         if (caps)
  1005.         {    char buf [100];
  1006.             register char *bp;
  1007.             strcpy (buf, *p);
  1008.             bp = buf;
  1009.             while (*bp)
  1010.             {    if ((*bp >= 'a') && (*bp <= 'z'))
  1011.                     *bp += ('A'-'a');
  1012.                 bp++;
  1013.             }
  1014.             chkerr (fprintf (anafile, "%s ", buf));
  1015.         }
  1016.         else chkerr (fprintf (anafile, "%s ", *p));
  1017.         lastanawords [p-anawords] = *p;
  1018.     }
  1019.     chkerr (fprintf (anafile, "\n"));
  1020.  
  1021.     return (1);
  1022. }
  1023.  
  1024. short exitcheck ()                                        /* return 0 to stop this anagram */
  1025. {
  1026.     short c;
  1027.     short stopped = 0;
  1028.     short keynum;
  1029.     char str [STRMAX];
  1030.  
  1031. #if 0
  1032.     while (kbhit ())
  1033.         if (getchar () == ' ')
  1034.             stopped = 1;
  1035. #endif
  1036.  
  1037.     if (! stopped) return (1);                    /* continue */
  1038.  
  1039.     stoptimer ();
  1040.     keynum = getline (
  1041.         "\nType STOP for the next anagram, QUIT to leave the program,\nor anything else to continue? ",
  1042.         "#stop", 0, "", 
  1043.         "Type STOP for the next anagram, QUIT to leave the program,\nor anything else to continue? ",
  1044.         quithelp, str);
  1045.  
  1046.     starttimer ();
  1047.     if (keynum == 0)                                        /* typed STOP? */
  1048.         return (0);                                                /* yes: stop */
  1049.     else return (1);                                        /* no: continue */
  1050. }
  1051.  
  1052. void getoutput ()
  1053.     /* GLOBAL OUTPUT: anafile, silent */
  1054. {
  1055.     char response [STRMAX];
  1056.     short done = 0;
  1057.     short keywd;
  1058.  
  1059.     printf ("Print to the screen, print to a file, or just count anagrams?\n");
  1060.     while (! done)
  1061.     {
  1062.         keywd = getline ("Type COUNT or a filename [or press RETURN for SCREEN]? ",
  1063.                         "#screen#count", 0, "screen",
  1064.                         "Type COUNT or a filename [or press RETURN for SCREEN]? ",
  1065.                         scf, response);
  1066.  
  1067.         if (keywd == 0)
  1068.         {                                                                    /* handle SCREEN */
  1069.             anafile = stdout;                                /* output to screen */
  1070.             done = TRUE;
  1071.         }
  1072.         else if (keywd == 1)                            /* handle COUNT */
  1073.         {
  1074.             anafile = NULL;                                    /* no output */
  1075.             done = TRUE;
  1076.         }
  1077.         else done = openoutfile (response);    /* see if we can output */
  1078.     }                                                                        /* end of loop 'til done */
  1079. }
  1080.  
  1081. short openoutfile (name)
  1082.     char *name;
  1083. {
  1084.     short keywd;
  1085.     char response [STRMAX];
  1086.     char *opentype;
  1087.  
  1088.     opentype = "w";                                            /* default open type */
  1089.     anafile = fopen (name, "r");
  1090.     if (anafile != NULL)                                /* got it? */
  1091.     {
  1092.         fseek (anafile, 0L, 2);                        /* reset to the end */
  1093.         if (ftell (anafile) != 0L)                /* position != 0?  (ie, not empty?) */
  1094.         {
  1095.             printf ("'%s' isn't empty.  Want to OVERWRITE or ADD to it [press RETURN to ADD]? ", name);
  1096.             keywd = getline ("", "#overwrite#add", 1, "add",
  1097.                                             "Please type OVERWRITE or ADD? ", overadd, response);
  1098.             if (keywd == 0)
  1099.                 opentype = "w";                                /* overwrite */
  1100.             else opentype = "a";                        /* add */
  1101.         }
  1102.         fclose (anafile);
  1103.     }
  1104.  
  1105.     anafile = fopen (name, opentype);        /* try to open it as desired */
  1106.  
  1107.     if (anafile == NULL)                                /* blew it? */
  1108.     {
  1109.         printf ("Sorry -- can't write to '%s'.  Try again?\n", name);
  1110.         return (0);
  1111.     }
  1112.     return (1);                                                    /* success */
  1113. }
  1114.  
  1115. short getline (    char *prompt,
  1116.             char *keywords, 
  1117.             short keyneeded,
  1118.             char *dft, 
  1119.             char *reprompt,
  1120.             short helpstate, 
  1121.             char *response)
  1122.                                                 /* INPUT: prompt */
  1123.                                             /* INPUT: keywords, like "#a#b#c" */
  1124.                                         /* INPUT: insist on a keyword? */
  1125.                                                     /* INPUT: default response */
  1126.                                             /* INPUT: repeated prompt */
  1127.                                         /* INPUT: help state */
  1128.                                             /* OUTPUT: response buffer */
  1129. {
  1130.     short done = 0;                                            /* flag to exit loop */
  1131.     short keyindex;                                            /* index of word in #string */
  1132.  
  1133.     printf ("%s", prompt);                            /* prompt 'em */
  1134.     while (! done)
  1135.     {
  1136.         gets (response);                                    /* get a line of input */
  1137.         if (strlen (response) == 0)                /* pressed RETURN? */
  1138.             strcpy (response, dft);                    /* yes: supply default */
  1139.  
  1140.         keyindex = keyfind ("#quit#help", response); /* see if it's a standard one */
  1141.         if (keyindex >= 0)                                /* found something? */
  1142.         {                                                                    /* handle QUIT or HELP */
  1143.             if (keyindex == 0)                            /* QUIT? */
  1144.             {
  1145.                 printf ("\n\n");
  1146.                 exit (0);
  1147.             }
  1148.             dohelp (helpstate);                            /* nope: HELP */
  1149.         }                                                                    /* end of standard keywords */
  1150.         else {                                                        /* not standard */
  1151.             keyindex = keyfind (keywords, response); /* look it up */
  1152.  
  1153.             if ( (keyindex >= 0)                        /* found something... */
  1154.                 || (! keyneeded) )                        /* ...or didn't need it */
  1155.                     done = TRUE;                                /* then we're satisfied */
  1156.         }                                                                    /* end of parsing response */
  1157.  
  1158.         if (! done) printf ("%s", reprompt);    /* about to loop? be pushy */
  1159.     }                                                                        /* end of loop 'til done */
  1160.  
  1161.     return (keyindex);                                    /* return keyword number or -1 */
  1162. }                                                                            /* end of getline() */
  1163.  
  1164. void dohelp (short state)
  1165.     
  1166. {
  1167.     printf ("\n");
  1168.     if (state == anahelp)
  1169.     {
  1170.         printf ("  Enter a word, name or phrase.  This program will print out all\n");
  1171.         printf ("  the combinations of words which can be made by rearranging the\n");
  1172.         printf ("  letters in the phrase you type in.  Very short phrases won't\n");
  1173.         printf ("  have many good anagrams, while long phrases may produce too\n");
  1174.         printf ("  many to sift through.\n");
  1175.     }
  1176.     else if (state == scf)
  1177.     {
  1178.         printf ("  You can print anagrams on the screen, store them in a file,\n");
  1179.         printf ("  or just count them without printing them anywhere.  To print\n");
  1180.         printf ("  them on the screen, type SCREEN or just press RETURN.  To count\n");
  1181.         printf ("  them, type COUNT.  To send them to a file, type the file's name.\n");
  1182.     }
  1183.     else if (state == overadd)
  1184.     {
  1185.         printf ("  You can replace the current contents of that document with the\n");
  1186.         printf ("  anagrams, or you can add them after the old contents.  To replace,\N");
  1187.         printf ("  type OVERWRITE.  To add, type ADD.\n");
  1188.     }
  1189.     else if (state == quithelp)
  1190.     {
  1191.         printf ("  When you press the space bar, it suspends anagram finding\n");
  1192.         printf ("  in case you want to quit.  To leave the program, type QUIT.\n");
  1193.         printf ("  To do another anagram, type STOP.  To resume, type anything else.\n");
  1194.     }
  1195.     else if (state == dict1)
  1196.     {
  1197.         printf ("  Enter the name of a dictionary file you'd like to use.  Usually,\n");
  1198.         printf ("  '%s' is used, but it doesn't seem to be available.\n", DFTFILE);
  1199.         printf ("  Any text-only file containing one word per line will do.\n");
  1200.     }
  1201.     else if (state == dict1dft)
  1202.     {
  1203.         printf ("  Enter the name of a dictionary file you'd like to use.  Press\n");
  1204.         printf ("  RETURN to use '%s', the default file.\n", DFTFILE);
  1205.     }
  1206.     else if (state == dict2)
  1207.     {
  1208.         printf ("  Enter the name of another dictionary file, or press RETURN\n");
  1209.         printf ("  if you don't want to enter any more.\n");
  1210.     }
  1211.     else if (state == helppage)
  1212.     {
  1213.         printf ("  Normally output to the screen rushes by unless you press the\n");
  1214.         printf ("  mouse button to freeze it or hit the space bar to suspend it.\n");
  1215.         printf ("  If you ask for 'paged' output, it'll stop after each screenful.\n");
  1216.     }
  1217.     else if (state == helppaused)
  1218.     {
  1219.         printf ("  Press RETURN to resume reading anagrams, STOP to do a different\n");
  1220.         printf ("  anagram, or QUIT to stop.\n");
  1221.     }
  1222.     else printf ("Sorry -- no help available here.\n");
  1223.  
  1224.     if (state != quithelp)
  1225.         printf ("  At any time, you can leave by typing QUIT.\n");
  1226.     printf ("\n");
  1227. }
  1228.  
  1229. short keyfind (words, oneword)
  1230.     char *words;
  1231.     char *oneword;
  1232. {
  1233.     short cindex;
  1234.     char wordcopy [STRMAX], lookupword [STRMAX];
  1235.     short wordnum;
  1236.  
  1237.     strcpy (wordcopy, oneword);
  1238.     clean (wordcopy);
  1239.     if (strlen (wordcopy) < 1) return (-1);    /* can't find nothing */
  1240.     strcpy (lookupword, "#");
  1241.     strcat (lookupword, wordcopy);            /* construct "#word" */
  1242.  
  1243.     cindex = findstr (words, lookupword);    /* look up "#word" in "#a#b... */
  1244.     if (cindex == -1)                                        /* not found? */
  1245.         return (-1);                                            /* say so */
  1246.  
  1247.     wordnum = 0;                                                /* initially we're at word zero */
  1248.     while (cindex-- > 0)                                /* loop up to "#" in "#word" in str */
  1249.         if (*words++ == '#')                            /* is there a "#"? */
  1250.             ++wordnum;                                            /* yes: that's another word */
  1251.  
  1252.     return (wordnum);
  1253. }
  1254.  
  1255. short findstr (master, sub)
  1256.     char *master;
  1257.     char *sub;
  1258. {
  1259.     short result = 0;
  1260.     short subsize = strlen (sub);
  1261.  
  1262.     while (*master)
  1263.     {
  1264.         if (strNequal (master, sub, subsize) == 1)    /* match? */
  1265.             return (result);                                /* yup: say where */
  1266.         ++result;
  1267.         ++master;
  1268.     }
  1269.  
  1270.     return (-1);                                                /* no match */
  1271. }
  1272.  
  1273. short strNequal (register char *s1,register char  *s2, register short size)
  1274. {    while (size--)
  1275.         if (*s1++ != *s2++) return 0;            /* not equal */
  1276.     return 1;                                                        /* equal */
  1277. }
  1278.  
  1279. long msec ()
  1280. {
  1281.     return (TickCount () * (1000/60));    /* convert from 1/60s to ms */
  1282. }
  1283.  
  1284. void starttimer ()
  1285. {
  1286.     startt = msec ();
  1287. }
  1288.  
  1289. void stoptimer ()
  1290. {
  1291.     endt = msec ();
  1292.     tottime += (endt-startt);
  1293. }
  1294.  
  1295. void inittimer ()
  1296. {
  1297.     tottime = 0;
  1298.     starttimer ();
  1299. }
  1300.  
  1301. void printtime ()
  1302. {
  1303.     stoptimer ();
  1304.     printf ("Computing time: %ld.%03ld seconds.  ",
  1305.                                     tottime/1000L, (tottime % 1000));
  1306. }
  1307.  
  1308. void chkerr (short i)
  1309. {
  1310.     if (i == EOF)
  1311.     {
  1312.         printf ("Sorry!  Error in writing to output file.  Perhaps your disk is full?\n");
  1313.         exit (0);
  1314.     }
  1315. }
  1316.